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This application claims priority to U.S. Provisional Patent Application Serial No. 
5 60/189,192 filed March 14, 2000, entitled "Difference Engine Method And Apparatus", 
the disclosure of which is incorporated herein in its entirety. 

Copyright 

A portion of the disclosure of this patent document contains material which is 
10 subject to copyright protection. The copyright owner has no objection to the facsimile 
reproduction by anyone of the patent document or the patent disclosure, as it appears in the 
Patent and Trademark Office patent files or records, but otherwise reserves all copyright 
rights whatsoever, 

15 Background of the Invention 

h Field of the Invention 

The present invention relates to the field of methods of manipulating data, and 
specifically to computer programs useful for the manipulation and analysis of data strings 
associated with one or more digital processors or peripheral devices, such as during 
20 processor debug analysis. 

2, Description of Related Technology 

RISC (or reduced instruction set computer) processors are well known in the 
computing arts. RISC processors generally have the fimdamental characteristic of 

25 utilizing a substantially reduced instruction set as compared to non-RISC (commonly 
known as "CISC") processors. Typically, RISC processor machine instructions are not 
all micro-coded, but rather may be executed immediately without decoding, thereby 
affording significant economies in terms of processing speed. This "streamlined" 
instruction handling capability fiirthermore allows greater simplicity in the design of the 

30 processor (as compared to non-RISC devices), thereby allowing smaller silicon and 
reduced cost of fabrication. 
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RISC processors are also typically characterized by (i) load/store memory 
architecture (i.e., only the load and store instructions have access to memory; other 
instructions operate via internal registers within the processor); (ii) unity of processor and 
compiler; and (iii) pipelining, 
5 In addition to the single RISC processor core described above, such cores may be 

used in conjunction with the same or other types of processor cores, whether as 
physically discrete components or as functions within a single die. For example, a 
plurality of similar or identical microprocessor cores may be used to fonn a multi- 
processor architecture adapted for parallel processing of data streams. Alternatively, a 

10 microprocessor core may be used with a digital signal processor (DSP) core on the same 
die, the DSP core performing high sampling rate operations requiring its unique 
architecture (such as FFT calculations or speech processing). 

In such multi-core environments (and in fact in other types of configurations), 
situations frequently arise wherein data may need to be converted from one format to 

15 another, or patterns within the data recognized and analyzed to identify useful functional 
relationships based thereon, or identify problems within the cores. For example, the 
comparison of the respective processor core stack traces for a debugging program has 
significant utility for such data manipulation and analysis. Similarly, the 
designer/programmer may wish to examine state information from the various cores 

20 during programming or design synthesis. 

One example of the foregoing problem is graphically illustrated in Table 1 below: 

Table 1 



Input 1 


Input 2 


Inputs 


Input 4 


A 


A 


A 


S 


R 


C 


C 


T 


B 


B 


B 


U 


C 


C 




V 


D 


D 


D 




B 


B 


B 
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Given the four input strings (i.e., Inputs 1-4) of Table 1, it might be desired to have the 
output formatted as follows: 



[1-3] 


A 


[1] 


R 


[2,3] 


C 


[1-3] 


B 


[1,2] 


C 


ri-31 


D 


[1-3] 


B 


[4] 


S 


[4] 


T 


[4] 


U 


[4] 


V 



Heretofore, algorithms adapted to format or analyze data have not been adapted and 
15 optimized for formatting such data derived from a plurality of inputs (such as a plurality 
of processor cores), and analyzing and recognizing patterns therein, especially in the 
context of multiprocessor core debug. For example, prior art UNIX-based systems 
having routines adapted for differencing two files (e.g., "diff") are generally useful only 
for comparing two files, and are not optimized for formatting or recognizing patterns or 
20 differences within "N" data inputs or files. 

When such formatting and analysis of the N inputs is optimized, the 
analysis/debug process is made more efficient as a whole, and more usefiil information 
can be readily extracted. Such is the case in debugging multiple parallel processor cores, 
such as the aforementioned RISC processors. Consider the example of a debugger, 
25 wherem the same program is run on two different processor cores simultaneously, and 
where the output of both programs is expected to be the same: 

SO go Try 

[1] queens.c!26: void Try(Integer I, Boolean *Q) { 
30 [1] Execution stopped at breakpoint. 

[2] queens.c!26: void Try(Integer I, Boolean *Q) { 
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[2] Execution stopped at breakpoint. 
SC> source 





LlJ> 


26 void Try(Integer I, Boolean *Q) { 




n 1 


// integer j — u. 


5 


[1] 


28 do { 




[1] 


,29 J++;*Q = False; 




[2]> 


26 void Try(Integer I, Boolean *Q) { 




[2] 


27 Integer J = 0; 




[2] 


28 do{ 


10 


[2] 


29 J++;*Q = False; 



Note that the output of both processors (identified as "[1]" and "[2]" in the code above) 
must be visiially inspected by the programmer to ensure that the two programs have 
arrived at the same function together. 

1 5 Furthermore, in the case where such formatting or pattern recognition is required 

in a repetitive or iterative fashion, inefficiencies such as the foregoing visual inspection 
requirement may be multiplied many times over. 

Based on the foregoing, there is a need for an improved method and apparatus for 
formatting such data from a plurality of outputs from one or more threads, and analyzing 

20 and recognizing patterns therein. Ideally, such method and apparatus would be adapted to 
format/analyze a plurality ("N") of different input sources, confirm the presence of 
similarities in the data associated with each source, and automatically identify the 
presence and location of differences therein. Such improved method would also be able 
to be reduced to an algorithmic or computer program representation for ready use in a 

25 variety of different hardware environments. 

Sununarv of the Invention 
The present invention satisfies the aforementioned needs by providing an improved 
method and apparatus for analyzing data strings, particularly in the multi-processor 
30 environment. 



-4- 



In a first aspect of the invention, an improved method for analyzing data 
comprising one or more strings or threads is disclosed. The method generally comprises 
initializing the data by building a symbol array; and finding differences within the data by 
analyzing various relationships witiiin the data strings, such as the existence of unique 
strings. 

In one exemplary embodiment, the method of initializing comprises creating an 
empty symbol table; creating a symbol array with at least one element for each input 
string; for each input string, determining whetiier the string is in the symbol table; and if 
the string is m the symbol table, then obtaining the symbol number of the string from the 
symbol table. 

In one exemplary embodiment, the method of fmding differences within the data 
comprises providing a plurality of inputs; processing inputs which do not share any 
strings with other inputs (i.e., "unique" inputs); evaluating the excluded set; identifying 
groups of contiguous strings which are identical in all inputs (i.e., fmding "shared 
chunks"); identifying groups that are in the same order in all of the inputs; identifying 
inputs witii strings before the first group; and fmding differences among the subset of 
inputs comprising such strings before the first group. 

In one exemplary embodiment, the method of processing unique inputs comprises 
clearing the symbol counts array; for each string in each input, incrementing tiie symbol 
count; for each string in each input, determining tiie symbol count for tiiat string; and for 
each input, adding the input to the excluded set. 

In one exemplary embodiment, the metiiod of fmding "shared chunks" comprises 
creating a list of anchors containing strings that occur at a predetermined frequency; for 
each input of each anchor, generating a new group and associating each anchor string 
with that group; for each input of each anchor, associating each string which precedes tiie 
group and which is identical in all inputs with the new group; and for each input of each 
anchor, associating each string which follows tiie group and which is identical in all 
inputs with the new group. 

In a second aspect of tiie invention, an improved computer program usefiil for 
analyzing and formatting data obtained from a plurality of sources and embodying tiie 
aforementioned metiiod(s) is disclosed, hi one exemplary embodiment, the computer 



program comprises an object code representation stored on the magnetic storage device 
of a microcomputer, and adapted to run on the central processing unit thereof. The 
computer program further comprises an interactive, menu-driven graphical user interface 
(GUI), thereby facilitating ease of use. 

In a third aspect of the invention, an improved apparatus for running the 
aforementioned computer program used for analyzing data strings associated with the 
design and/or operation of processors is disclosed. In one exemplary embodiment, the 
system comprises a stand-alone microcomputer system having a display, central 
processing unit, data storage device(s), and input device. The system is adapted to take a 
plurality of data inputs from various external processor devices and format and analyze 
the inputs in order to confirm similarities and identify differences therein, and display the 
results of such analysis to the user (programmer). 

In a fourth aspect of the invention, an improved method of designing a processing 
device adapted to run at least one software process thereon is disclosed. The method 
generally comprises generating a design for the processing device; running at least a 
portion of the software process a first time; obtaining a first output firom the running 
process; modifying the design; running the software process a second time; obtaining a 
second output from the running process; and identifying differences within the first and 
second outputs. 

In a fifth aspect of the invention, an improved method of evaluating the operation 
of a plurality of software processes running on respective ones of a plurality of digital 
processors is disclosed. The method generally comprises generating a first data string 
using a first of the plurality of software processes; generating a second data string using a 
second of the plurality of software processes; inputting the first and second data strings 
into a debug software process; analyzing tiie first and second data strings using the debug 
process; and evaluating the operation of the processes based at least in part on the act of 
analyzing. In one exemplary embodiment, the debug process is run on a microcomputer, 
and is adapted to confirm matching portions of the applications running on the digital 
processors, and identifying differing portions to alter the programmer to possible bugs in 
the design(s). 



Brief Description of the Drawings 
Fig. 1 is a logical flow diagram illustrating the general methodology of analyzing 
data having strings or threads employed in the difference engine of the present invention. 
Fig. 2 is a logical flow diagram illustrating one exemplary embodiment of the 
5 method of initiaUzing the difference engine of the present invention. 

Figs. 3 is a logical flow diagram illustrating one exemplary embodiment of the 
method of finding differences in the input data strings used in the difference engine of the 
present invention. 

Fig. 4 is a logical flow diagram illustrating one exemplary embodiment of the 
10 method processing unique inputs employed in the difference engine of the present 
invention. 

Fig. 5 is a logical flow diagram illustrating one exemplary embodiment of the 
method of non-group comparison employed in the difference engine of the present 
invention. 

15 Fig. 6 is a logical flow diagram illustrating one exemplary embodiment of the 

methodology of separately printing strings according to the present invention. 

Fig. 7 is a logical flow diagram illustrating one exemplary embodiment of the 
methodology of printing group contents according to the present invention. 

Fig. 8 is a logical flow diagram illustrating one exemplary embodiment of the 
20 methodology of finding shared chunks according to the present invention. 

Fig. 9 is a functional block diagram of one exemplary embodiment of a computer 
system useful for running a computer program embodying the method of Figs. 1-8. 

Fig. 10 is a functional block diagram of the computer system of Fig. 9, including 
data interfaces to a plurality of processors being debugged. 

25 

Detailed Description 
Reference is now made to the drawings wherein like numerals refer to like parts 
throughout. 

As used herein, the term "processor" is meant to include any integrated circuit or 
30 other electronic device capable of performing an operation on at least one instruction 
word including, without limitation, reduced instruction set core (RISC) processors such 
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as the ARC™ user-configurable core manufactured by the Assignee hereof, central 
processing units (CPUs), and digital signal processors (DSPs). The hardware of such 
devices may be integrated onto a single piece of silicon ("die"), or distributed among two 
or more dies. Furthermore, various functional aspects of the processor may be 
5 implemented solely as software or firmware associated with the processor. 

As used herein the term "string" refers to the fundamental unit of input for 
purposes of comparison; e.g., a sequence of ASCII characters of arbitrary or fixed length. 

As used herein the term "group" refers to a contiguous sequence of strings 
containing at least one anchor and for which the entire sequence has been determined to 
1 0 exist in all inputs currently under consideration. 

As used herein the term "chunk" refers to a contiguous sequence of strings. 



Description of Method 

Referring now to Fig. 1, the method of analyzing and formatting data obtained 

15 from a plurality of data sources according to the present invention is described. While 
the following discussion is cast in terms of the logical flow of a computer program 
adapted to perform the method of the invention when run on a microcomputer system 
such as that of Fig. 9, it will be appreciated that the methods described herein may be 
embodied in other forms including, for example, hardware (e.g., logic circuits disposed 

20 within an integrated circuit such as an FPGA or ASIC) specifically adapted to provide the 
desired functionality. 

As illustrated in Fig. 1, the method 100 generally comprises the steps of 
initializing the data obtained from the data sources (step 102), and finding (and printing, 
as applicable) the differences between the various inputs (step 104). 

25 Figure 2 describes the initiaUzation step 102 in greater detail. As shown in Fig. 2, 

the process of initialization 102 comprises first creating an empty symbol table and 
setting the symbol number to zero per step 202. As used herein, the term "symbol 
number" refers to that number assigned to a corresponding distinct input. Next, in step 
204, a symbol array with one element for each input string is created for each input, and 

30 the string number is set to zero. In step 206, the presence of the string in the symbol table 
is determined for each string of each input. If a string is not in the symbol table per step 
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206, the symbol number is incremented, and the string is associated with a symbol 
number in the symbol table per step 208. In step 210, the symbol number associated with 
the string (however obtained) is obtained from the symbol table, the line number is 
incremented, and the symbol array (line number) is set to the value of that symbol 
5 number. This process is repeated iteratively for each string in each input, using the 
foregoing steps, until all inputs have been analyzed, and the method 200 returns per step 
212. 

Referring now to Fig 3, one embodiment of the method 300 of finding differences 
according to step 104 of Fig, 1 is described. First, the inputs are provided in step 302. 

10 Next, in step 304, inputs that do not share any strings with other inputs are processed. 
These inputs are termed "unique." Fig. 4 (described below) illustrates one embodiment 
of the method of processing the mput data according to the invention, although other 
approaches may be substituted dependent on the needs of the particular appUcation. 

In step 306, the excluded set is examined to determine if it is empty. If not empty, 

15 the differences are identified using the "included" set as inputs per step 308. As used 
herein, the term "included" and "excluded" refer to data or elements which meet a 
predetermined criterion (and are hence "included" in a set of elements also meeting that 
criteria), or do not meet the predetermined criteria (and are hence "excluded"). In the 
present example, the criterion is whether the excluded set has any members (i.e., empty), 

20 although it will be recognized that other criteria may be specified in place of or in 
addition to the aforementioned criterion. For each element of the excluded set (step 310), 
the element is printed per step 312. When this process is completed, the program returns 
per step 313. 

If the excluded set is empty per step 306, groups of contiguous strings which are 
25 identical in all inputs are identified per step 314. These are referred to as "shared 
chunks;" shared chunks are described further with respect to Fig. 8 below. In step 316, 
the order of the groups or "shared chunks" is examined to determine if the groups are 
present in the same order in all inputs. If not, the method 300 returns per step 313. If so, 
the method 300 continues with step 318 of Fig. 3. 
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In step 318, it is determined whether any groups exist. If not, a line-by-line 
comparison of inputs is performed per step 320; non-group comparison is described 
fiirther with respect to Fig. 5 herein. The method 300 then returns per step 33 1 . 

If groups do exist per step 318, it is next determined if any of the inputs have 
5 strings before the first group (step 322); if so, then the differences among that subset of 
the inputs which consists of the strings which precede the first group in each input is 
found per step 324. For each group (step 326), the contents of the group are then printed 
per step 328. In step 330, the differences among that subset of the inputs which consists 
of the strings between the present group and the next group (or end of input) are 
10 determined. After these steps 328, 330 have been performed for each group, the method 
300 returns per step 33 1 . 

Referring now to Fig 4, one embodiment of the method of processing unique 
inputs is described. The method 400 is entered with the aforementioned inputs in step 
402. Next, the symbol counts array is cleared in step 404. For each string (step 408) in 
15 each input (step 406), the symbol count is incremented per step 410. When completed 
for all strings of all inputs, the method 400 proceeds to step 412 of Fig. 4. 

For each string (step 414) in each input (step 412), the symbol coxint is checked 
per step 416. If the count is greater than a predetermined number (1 in the present 
embodiment), the input is added to the excluded set per step 418. Each input is then 
20 added to the included set in step 420, until all inputs are added and the method return per 
step 421. 

Referring now to Fig. 5, one embodiment of the method 500 of comparing non- 
groups according to the present invention is described. The method 500 is entered with 
the given inputs per step 502. For each input (step 504), the percentage of strings in the 

25 input that also appear in other inputs is determined in step 506. If the percentage is 
greater than a predetermined tolerance (step 508), the input is added to the included set in 
step 510. If not, the input is added to the excluded set per step 512. When completed, 
the method 500 proceeds to step 514 of Fig. 5. 

As illustrated in Fig. 5, the printed set is first emptied per step 514. Next, the line 

30 is set to null per step 516. For each included element (step 5 1 8), it is determined whether 
the included element is in the printed set per step 519. If yes, the method returns to step 



-10- 



518. If not, the included element is examined to determine if it contains any unprinted 
strings per step 520. If not, the included element is removed from the included set per 
step 522. If so, it is next determined whether the line is null in step 524. The Une is then 
set to the next string from the included set in step 526. If the line is not null, it is 
compared with the next string from the input per step 528. If it is equal, then the included 
element is added to the printed set per step 530. If the Ime is greater than or less than 
(i.e., not equal) the next string, the method returns to step 518 previously described. After 
the foregoing process is complete, the printed set is examined to determine whether it is 
empty per step 532. If the answer to the query of step 532 is "no", the printed set and line 
are printed per step 533, and the process is repeated by returning to step 516. If the 
answer is "yes", all included sets are then examined in step 538 to determine whether any 
have more strings. If not, the method contmues to step 540 of Fig. 5, described below. If 
so, the method returns to step 514 for additional processing. 

Referring again to Fig. 5, each element of the exclude set (step 540) is printed in 
step 542. The method 500 subsequently returns per step 544 upon completion. 

Referring now to Fig. 6, one embodiment of the method 600 of separately printing 
one or more parameters associated with each input string according to the present 
invention is described. The method 600 is entered using the given input in step 602. It is 
noted that in the illustrated embodiment, the input of step 602 is a sub-sequence of one of 
the original input data streams with the identity of the original stream. For each string in 
the input (step 604), the input identification (ID) and string is printed per step 606. 
When the parameters for all strings in the input have been printed, the method returns per 
step 608. 

Referring now to Fig. 7, one embodiment of the method 700 of printing the 
contents of a group according to the present invention is described. The method 700 is 
entered using a given group in step 702. As used herein, a group refers to a sequence of 
strings which occur in all of the original input data streams. For each string in the given 
group (step 704), the ID sequence for ail inputs and strings is printed per step 706 (i.e., 
the information identifying each of the inputs and the string currently under consideration 
is printed). After completion, the method 700 returns per step 708. 



Referring now to Fig, 8, one embodiment of the method 800 of finding shared 
chunks according to the present invention is described. The method 800 is entered with 
the given inputs in step 802. Next, a list of anchors containing strings that occur exactly 
once in every input are created in step 804. As used herein, the term "anchor" refers to 
strings that occur exactly once in every input. For each input (step 808) of each anchor 
(step 806), a new group is created, the anchor string is marked as belonging to the new 
group per step 810. Next, in step 812, it is determined whether the string before the 
group (i.e., before the sequence of strings shared among all the inputs) is identical in each 
of the inputs. In step 816, the string is marked or otherwise identified as belonging to 
that group, and the string is removed fi-om list of anchors. When completed, the string 
immediately after the group is examined to determine if it is identical in each of the 
inputs per step 818. The string is marked as belonging to that group, and the string is 
removed from the list of anchors per step 820. 

It is noted that the use of a symbol table and sequentially allocated symbol 
numbers in conjunction with the foregoing method is not required, but it does provide a 
significant performance improvement during the comparison phases. Specifically, such 
performance improvements may be obtained through the use of a symbol table with 
sequentially allocated symbol identifiers (IDs), as well as provisions for maintaining 
information about which inputs contain a given line and frequency of occurrence thereof 
In one embodiment, the method of unplementing the aforementioned symbol table 
comprises first scanning through each input to build the table, and then replacing each 
input line with its associated symbol. The various line comparisons are therefore reduced 
to simple comparisons of the symbol's ID. Other methods of producing a symbol table 
may also be substituted however. 

Furthermore, it is noted that by counting how often each symbol occurs, and in 
which inputs, the need to actually scan through the individual inputs looking for matches 
may be reduced or even eliminated. 

It will also be recognized that the methodology previously described herein with 
respect to Figs. 1-8 is recursive in nature. Specifically, as illustrated in the following 
exemplary source code (and Appendix I hereto), a recursive call to the iterator 
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"each_output" on the subset_ranges is utilized to identify useful differences and 
similarities among the various subsets of data: 

// Exclude some because they are totally unique. Run the 

// algorithm on the subset, and at the end, print the unique ones. 

// Fashion arguments that represent the subset of items. 

Input_infos subset_ranges; 

for (int i = 0; i < input_count; i++) 

if (!exclude_set.contains(i)) subset_ranges.add(infos[i]); 

if (subset rangesxountO) each_output(yield,subset_ranges); 

AAAAAAAAAAAA 

for (int i = 0; i < input_count; i++) 
if (exclude_set.contains(i)) 
print_separately(yield,i,infos[i]); 

Furthermore, it will be recognized that due to the recursive nature of the 
invention, the terms "all inputs" and "all strings" generally refer only to those inputs or 
strings under analysis or consideration during a given recursion. 

Appendix I hereto provides exemplary code for the functionality illustrated in 
Figs. 1-8 herein, implemented in a dialect of C++ which includes iterators. As is well 
known, iterators are C++ intrinsic operators implemented in many C++ compilation 
systems such as the High C/C++ compilation system of the Assignee hereof However, 
the foregoing methods may be readily reduced to source code listings in any other useful 
higher level programmmg language, and subsequently compiled, by one of ordinary skill 
m the computer programming arts. 

Example of Method 

The following example illustrates the foregoing methodology in detail, in the 
context of the four data inputs of Table 1 above. 
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1 . First, each input is examined and any strings which occur only once in that input are 
identified. These strings are designated as 'unique'. 

2. Next, the unique strings in each input are compared to determine which ones are also 
unique in all other inputs. If any input does not share any unique strings with any 
other input, that input is set aside to be displayed separately and repeat the test with 
the remaining mputs. If there are no shared imique strings, Step 7 below is next 
performed. 

3. For the data of Table 1, Input 4 would be set aside, leaving the following: 



Table 2 



Input 1 


Input 2 


Input 3 


A 


A 


A 


R 


C 


C 


B 


B 


B 


C 


C 




D 


D 


D 


B 


B 


B 



Note that strings A and D were identified as "unique." 

4. Next, the inputs are split into a series of matching and non-matching chunks; where 
the matching chunks contain the shared unique strings; and the non-matching chunks 
contain the rest of the input. In the case of the data provided in Table 1: 
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Tables 
Input 1 Input! 

Match A A 



Input 3 

A 



R C C 

Non-match B B B 

C C 

Match D D D 

Non-match B B B 

4. Next, the inputs are examined to ensure that the matching chunks occur in the same 
order in every input. If they do not, a string-by-string comparison (discussed below) 
is performed. 

5. Next, the matching chunks are "grown" by absorbing surrounding strings which are 
identical in all inputs. Such strings are also removed from the non-matching chunks. 

a. Each matching chunk is grown backwards by comparing the string at the end of 
the immediately preceding non-matching chunk in each input. If that string is 
identical in all inputs, it is removed from each input's non-matchmg chunk and 
prepended to the matching chunk. If the preceding non-matching chunk is 
completely emptied (in all inputs), it is deleted, and the now adjacent preceding 
matching chunk is merged into this chunk and the foregoing step repeated with 
the new merged chunk. The process is ceased when the beginning of any input is 
reached, or when the preceding string is not identical in every input. 

b. Similarly each matching chunk is grown forward by comparing and appending 
the first string of the immediately following non-matching chunk until the end of 
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an input is reached or the following string is not identical in all inputs. In the 
preceding example of Table 1: 

Table 4 

Input 1 Input 2 Input 3 

Match A 

R C C 

Non-match B B B 

C C 



Match D 

B 



Next, the chunks are stepped through in order as they appear in the inputs. If a chunk 
is a matching chunk, its contents are displayed, along with an indication of which 
input streams contained it. If it is a non-matching chunk, the foregoing method is 
applied recursively fi:om Step 1, taking the contents of each mput's non-matching 
chunk as the entire contents of that input. 

In the example of Table 1 : 

Display: [1-3] A and recur, with the new inputs being: 
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Table 5 

Input 1 Input 2 Input 3 

R C C 

B B B 

C C 

Display: [1-3] D 
Display: [1-3] B 

5 

7. The inputs which were set aside in step 2 are now stepped through successively, the 
contents of each being displayed along with an identification of which input is being 
displayed. In the Table 1 example: 

Display: [4] S 

10 Display: [4] T 

Display: [4] U 

Display: [4] V 

String-by-String Comparison 

1 . Each input is scanned, counting the percentage of strings that also occur in at least 

1 5 one other input. If that percentage exceeds a pre-set threshold, that input is set aside 
to be printed separately. 

2. The first string of the first input is compared with the first string of each of the other 
inputs, separating the inputs into those for which the first string matches, and those 
which don't. 

20 3. The identities of the matching inputs, and the string, are displayed. 

4. Steps 2 and 3 are next repeated for the inputs which did not match. 

5. Steps 2 through 4 are then repeated for each string until all of the inputs are 
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exhausted. 

6. Next, the inputs which were set aside in step 1 are stepped through successively. For 
each input, its strings are stepped through individually and sequentially, printing the 
input identification and the string. 

5 It will be recognized that while the foregoing example and description with respect to 
Figs. 1-8 herein are cast in terms of a specific series of steps for accomplishing the 
desired result (i.e., analysis and formatting of the input data), various permutations of this 
series of steps, including substitution and/or addition of other steps, may be used 
consistent with the invention disclosed herein. For example, it was previously noted that 

10 the use of a symbol table may in certain circumstances enhance the performance of the 
method as a whole. However, such symbol table is not essential to the practice of the 
invention. 

Accordingly, the scope of the disclosed invention should be determined by the 
claims appended hereto, without respect to specific embodiments or limitations presented 
1 5 within the foregoing discussion. 

Applications 

There are many applications where it is useful or even necessary to determine the 
differences among multiple output streams. Quickly locating the areas of difference and 
20 readily identifying which streams differ and which agree at each point of difference can 
greatly speed the development and debugging processes. In some cases, the goal is to 
modify the process or device which generates the output being analyzed; the analysis 
verifies that the output remams within the expected and desired parameters. In other 
cases, the modifications are expected to change the outputs in ways that indicate some 
25 improvement in the generating mechanism. 

As an example of the foregoing, the present mvention may be applied to compare 
outputs of multiple design simulations having varying parameters in order to assess the 
relative merits of each design. 

Another application of the invention is for debugging multi-processor hardware. 
30 Specifically, in one aspect, the same task can be executed on each processor with the 
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outputs compared by this method. Potential hardware and software integration problems 
may be discovered and located by examining any areas of difference among the outputs. 
Similarly, when debugging simulated processor functions in a multi-processor 
environment, the method of the present invention may be used to compare the outputs of 
5 each of the simultaneous processes. Any differences readily identify which processes 
differ and in what areas of the output. 

When testing a processor or ASIC design and comparing outputs from multiple 
simulation types (gate, switch, logic, etc.), the present invention may be used to identify 
differences among the outputs that are indicative of areas of the design which may 
1 0 require refinement or verification. 

As yet another application, the present invention may be used to compare multiple 
instances of a netlist for different optimizations of a hardware design (e.g., through a 
synthesizer). 

It is also noted that certain high-reliability systems (e.g., early design of the primary 

1 5 on-board computer systems in the space shuttle) rely upon concurrent computation with a 
comparison of the results and some sort of "voting" scheme to automatically determine 
when a processor may be producing unreliable results. Such high-reliability systems were 
used, for example, in the early design of the primary on-board computer systems in the 
space shuttle. These systems often use different algorithms in some of the concurrent 

20 processes so that any flaw in the basic algorithm is likely to be caught during the voting 
stage. During the development and debugging of such systems, the methodology of the 
present invention advantageously provides a mechanism for readily highhghting any 
differences in the outputs among the various of such concurrent processes. Hence, real- 
time analysis of the outputs of such systems may be accompUshed. 

25 As yet another application of the invention, divergent modifications to a single source 
code base may be compared. Specifically, the methodology of the invention will facilitate 
identification of the areas where each modified source differs from the original base (and 
from the other modifications), as well as the areas where the modifications affected the 
same portions of code. This functionality is useful for either choosing between the 

30 various modification sets, or re-integrating them into a single merged source base. 
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It will also be recognized that the methodology of the present invention may be used 
when migrating a given program from one processor to another. For example, a 
programmer or designer desiring to "move" an application from one core to another can 
use the debugger disclosed herein to debug the same program on both processors at the 
same tune. Furthermore, the debugger may be used to display data at different points in 
each program which should be identical to that in the other program. This identical 
relationship (or lack thereof) can be automatically confirmed using the methodology of 
the present invention. 

Apparatus for Implementing Methodology 

Referring now to Fig. 9, one embodiment of a computing device capable of 
implementmg the data input analysis and formatting methods discussed previously herein 
with respect to Figs. 1-9 is described. The computing device 900 comprises a 
motherboard 901 having a central processing unit (CPU) 902, random access memory 
(RAM) 904, and memory controller 905. A storage device 906 (such as a hard disk drive 
or CD-ROM), input device 907 (such as a keyboard or mouse), and display device 908 
(such as a CRT, plasma, or TFT display), as well as buses necessary to support the 
operation of the host and peripheral components, are also provided. The method of Figs. 
1-9 are embodied in the form of an object code representation of a computer program and 
stored in the RAM 904 and/or storage device 906 for use by the CPU 902 during 
analysis, the latter being well knovra in the computmg arts. Alternatively, the computer 
program may reside on a removable storage device (not shown) such as a floppy disk or 
magnetic data cartridge of the type also well known in the art. The user (not shown) 
analyzes the data input from the various data sources by inputting initiating operation of 
the computer program via the program displays and the input device 907 during system 
operation. Alternatively, the system may be configured to automatically accept (and store 
if desired) the various data inputs and run the computer program when sufficient data 
exist, or on a periodic or ongoing basis. Many such alternative are possible, each being 
well within the skill of the ordinary practitioner. Analyses and/or formatted data 
generated by the program are stored in the storage device 906 for later retrieval, 
displayed on the graphic display device 908 for viewing by the user, or output to an 
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external device such as a printer, data storage unit, other peripheral component via a 
serial or parallel port 912 if desired. 

Fig. 10 illustrates one embodiment of the apparatus of Fig. 9 in data communication 
with tv^o processor devices (e.g., two RISC cores, one RISC core and one DSP function, 
5 one RISC core and one ASIC macro-function, etc.). It will be recognized that while two 
processing devices 1002, 1004 are illustrated in the system 1000 of Fig. 10, the present 
invention may be utilized in conjunction with any number (N) of different software 
processes or data sources disposed on any number of separate devices. The system 1000 
further includes two data interfaces 1010, 1012 adapted for two-way data communication 
10 between the apparatus 900 and the respective data processors 1002, 1004. 

Computer Program 

A computer program for implementing the aforementioned methods of data analysis 
is now described. In one exemplary embodiment, the computer program comprises an 

15 object ("machine") code representation of a C^ source code listing implementing the 
methodology of Figs, 1-8 herein. While C^ language is selected for the present 
embodiment, it will be appreciated that other programming languages may be used, 
including for example VisualBasic™, Fortran, and C^. The object code representation of 
the source code listmg is compiled and disposed on a media storage device of the type 

20 well known in the computer arts. Such media storage devices can include, without 
limitation, optical discs, CD ROMs, magnetic floppy disks or "hard" drives, tape drives, 
or even magnetic bubble memory. The computer program further comprises a graphical 
user interface (GUI) of the type well known in the programming arts, which is 
operatively coupled to the display and input device of the host computer or apparatus on 

25 which the program is run (described above with respect to Fig. 9). 

In a second embodiment, the computer program of the invention comprises an 
assembly language/micro-coded instruction set disposed within the embedded storage 
device, i.e. program memory, of a digital signal processor (DSP) or microprocessor. 

While the above detailed description has shown, described, and pointed out novel 

30 features of the invention as applied to various embodiments, it will be understood that 
various omissions, substitutions, and changes in the form and details of the device or 
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process illustrated may be made by those skilled in the art without departing from the 
invention. The foregoing description is of the best mode presently contemplated of 
carrying out the invention. This description is in no way meant to be limiting, but rather 
should be taken as illustrative of the general principles of the invention. The scope of the 
invention should be determined with reference to the claims. 
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Copyright © 1999-2001 MetaWare Software Inc. All rights reserved. 

V 

// This module aggregates the text lines of an arbitrary number of 
// input streams, comparing each with the other for similarities 
// and joining up common lines where possible. 

tinclude "globals.h" 
#include "textagg.h" 
#c_include . /ui /report . h" 
#include " . . /utility/symbols . h" 
#c_include . /utility/string. h" 
#c_include . /utility/longvect . h" 
#c_include " . . /utility/ptrvecto . h" 
#c_include " . . /utility/set . h" 
#c_include . /common/boolean. h" 
#c_include <string.h> 

/* $Log: /sol/src/dbg/sc/cll/textagg. ccv $ 

Aggregator : : -Aggregator ( ) { } 
struct Real_aggregator : Aggregator { 
Real_aggregator (Symbols ^_sym) { 

set_tolerance (20) ; 

delete_st = _sym = 0; 

St = _sym ? _sym : Symbols : :make_Symbols () / 
} 

bool delete__st; 

override void set^tolerance (unsigned percent) { 
if (percent <= 100) tolerance = percent; 
} 

override -Real_aggregator ( ) { 
if (delete_st) delete st; 
} 

unsigned tolerance; 
Symbols *st; 

typedef const char -^OVT; 

override void each_output (Inputs ^inputs) -> (OUT, OUT); 
// private stuff: 
Inputs * inputs; 
typedef Uint_set Set; 
#define NOINLINE virtual 
NOINLINE void print_set (Set ^s) { 
print f ("set: ") ; 

for I <- s.each^element 0 do printf("%d ",!); 

printf ("\n") ; 

} 

// Data computed during analysis and shared among the routines, 
int global_input_count; 
ExtendingLongVector *symindex_vectors; 

// We build a table of these that store the info for each input, 
struct Input_inf o { 

int lo,hi; //We need int so we can have -1. 

Input* input; // Input for this index. 
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// Vector that maps 0..#lines-l to the symbol indices. 

ExtendingLongVector * symindex_vector ; 

}; 

typedef Extending_Vector__of <Input__inf o> Input_infos; 
void each_output (Input_infos &infos) -> (OUT, OUT); 
void initialize ( Inputs ^inputs, Input_infos &Rs) ; 
int depth, global_maxlc, global_minlc; 

void non_group_comparisons (Input__infos &infos, int itiaxlc) -> 
(OUT, OUT) ; 

void print_separately (int i, Input_info &R) -> (OUT, OUT); 
}; 

Aggregator * Aggregator : :make_Aggregator (Symbols *syms) { 
return new Real_aggregator (syms) ; 
} 

void Real_aggregator :: initialize (Inputs "^^inputs, Input_infos &Rs) { 
inputs = _inputs; 

// Phase 1. Collect all the lines and assign them a unique Symbol 
// number for later comparison. Along with each line record which 
// stream it comes from. 
global_input_count = 0; 

for I <- inputs . each_input ( ) do global_input_count++; 
void failed {) { 

// Perhaps set a return value? Boolean by var? 

} 

if (global_input_count == 0) { failedO; return; } 

// Store the Symbol indices for each string in each input. 

symindex_vectors = new ExtendingLongVector [global_input_count ] ; 

if (symindex_vectors ==0) { failed () ; return; } 

int icnt = 0; 

for I <- inputs . each_input ( ) do { 

ExtendingLongVector &vec = symindex_vectors [icnt ] ; 

Rs [icnt ]. input = I; 

Rs [icnt ] . symindex_vector = &vec; 

int lineno = 0; 

if (delete_st FALSE) 

for S <- I . each__line_symbol ( ) do vec [lineno-f+] = S; 
else for line, persistent <- I . each_line ( ) do { 

Symbol LS = st . get_last_symbol ( ) ; 

Symbol S = st .make_symbol (line) ; 

if (S > LS && dbg) printf ("Symbol %d is %s\n", S, line) ; 

vec [lineno++] = S; 

} 

icnt++; 
} 



void Real_aggregator : :each_output (Inputs ^inputs) -> (OUT, OUT) { 
dbg && printf ( "entering aggregatorXn" ) ; 
Input_infos Rs; 
initialize (inputs, Rs); 
if (global_input_count == 0) return; 

global_minlc = global_maxlc =0; 

for (int i ^ 0; i < global_input_count ; i++) { 
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Input_info &R ^ Rs[i]; 

R.lo = 0; R,hi = syiuindex_vectors [i] . count {) -1 ; 
global_inaxlc = _max (global_maxlc, R.hi-R. lo+l) ; 
global_ininlc = _min (global_minlc, R. hi-R. lo+l) ; 
} 

depth = 0; 

if (global_maxlc) { 

if (global_input_count 1) 

// This will effectively just print the stuff. 
non_group_comparisons (yield, Rs, global_maxlc) ; 
else each_output (yields Rs) ; 
} 

delete [] symindex_vectors ; 

dbg && print f { "exiting aggregator\n") ; 

} 

void Real_aggregator : :each_output (Input_infos &infos) -> (OUT, OUT) { 
depth++; 

int input_count = infos . count () ; 
if (dbg) { 

printf (" [%d] Aggregate. ic %d Input_inf os : depth, input^count) ; 
for (int i ^ 0; i < input_count; i++) 

printf ("\t [%d,%d] ",infos[i] .lo,infos[i] .hi) ; 
printf ("\n") ; 
} 

int maxlc = 0, minlc = 99999999; 

// To compute whether a line appears in all inputs, we ust 

// input s__for_syinbol to record the last input for which it appeared. 

struct Syiti_info { 

int inputs_f or_syinbol; // Used to figure out if all inputs 
// have this line. 

unsigned occurrences; // tells which lines occur singly. 

unsigned line_index; // Where the line occurs, for the last 

input . 

}; 

Extending_Vector_of <Sym_inf o> sym_inf o ; 
for (int i = 0; i < input_count; i++) { 
Input_info &R = infos [i]; 

LongVector &vec = *inf os [i] . symindex^vector ; 
for (int line = R.lo; line <= R.hi; line++) { 

Symbol S = vec[line]; 

Sym_info &si = sym_info[S]; 

// Need to initialize this field. 

si . inputs__f or_symbol = -1; 

si . occurrences++ ; 

} 

} 

// Exclude from consideration an input stream that has no matches 

with 

// any other input stream. Let it be printed alone. This prevents 
// a rogue input stream from killing the matching algorithms, which 

are 

// based on unique lines across all inputs. 
Set *exclude_set Set : :make_Set ( ) ; 
for (int i = 0; i < input__count ; i++) { 
Input_info &R = infos [i]; 
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LongVector &vec = *infos [i] . symindex__vectcr ; 
int dups = 0; 

for (int line = R.lo; line <= R.hi; line++) { 
Symbol S = vac [line]; 
Sym_info &si = syin_inf o [S] ; 
if (si . occurrences > 1) dups++; 
} 

if (dups ==0) { 

dbg && printf ("input %d matches no one else\n",i); 

exclude_set . add_element (i ) ; 

} 

} 

if (exclude_set . count_eleinents 0 ) { 

dbg printf ( "exclude set is non-empty\n" ) ; 

// Exclude some because they are totally unique. Run the 

// algorithm on the subset, and at the end, print the unique ones. 

// Fashion arguments that represent the subset of items. 

Input_infos subset_ranges ; 

for (int i = 0; i < input_count; i++) 

if ( ! exclude_set . contains (i) ) subset_ranges . add (infos [i] ) ; 
if (subset_ranges . count ( ) ) each_output (yield, subset_ranges ) ; 
for (int i = 0; i < input_count; i++) 

if (exclude_set . contains (i) ) 

print_separately (yield, i, infos [i] ) ; 
delete excludG__set ; 
depth — ; 
return; 
} 

delete exclude_set; 

// If we get here, each input stream has a line that exists in at 
least 

// one other input stream. Still, there may be no line that^s 
unique . 

Set ^^absorbeds = new Set'*' [input_count ] ; 
Set *symbols_f or_this_iteration = Set : ;make_Set ( ) ; 
for (int i = 0; i < input_count; i++) { 
Input_info &R = inf os [i] ; 

LongVector &vec = "^inf os [i] . symindex_vector; 

absorbeds[i] = Set : :make_Set ( ) ; 

for (int line = R.lo; line <= R.hi; line++) { 

Symbol S = vec[line]; 

Sym_info &si = sym_info[S]; 

// Give this field a chance to advance once per iteration, 
if (si. input s_for_symbol i-1) si . input s_for_symbol++; 
si . line_index ^ line; 

symbols_for_this_iteration.add_element (S) ; 
} 

maxlc = _max (maxlc, R. hi-R. lo+l ) ; 
minlc = __min (minlc, R.hi-R. lo-f 1 ) ; 
} 

// Next identify unique lines in all streams. These are lines that 
// appear just once in all streams. So, find lines with # 
occurrences 

// the same as the number of streams, and which are in all the 
streams . 

unsigned last_symbol = st . get_last_symbol () ; 
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bool occurs__uniquely (Symbol S) { 

return sym_info [S] . input s_for__syinbol === input__count-l 
sym_info [S] . occurrences =^ input^count; 

} 

if (dbg) { 

printf ( "syms for this iteration; input count %d: " , input_count ) 
for S <- syiubols_f or_this_iteration . each_element 0 do 
printf ("%d <%d, %d> ",S, 

sym_info[S] . input s_for_symbol, 

sym_info[S] .occurrences) ; 
printf ("\n") ; 

for S <- syinbols_f or_this_iteration. each_element () do { 
if (occurs_uniquely (S) ) { 

printf ( "Occurs uniquely at last %d: %s\n", 
sym_info [S] . line_index, st . to_string (S) ) ; 

} 

} 

} 

// Now take each unique line and extend back and forth to create 
// group of matching lines surrounding the unique line, 

unsigned find_line_in (unsigned input, Symbol S, unsigned hint) { 
// find unique S in input near line hint. 
LongVector &L = *infos [input] . syinindex_vect or ; 
if (hint < L. count [) && L[hint] == S) return hint; 
unsigned first_nearby = hint > 5 ? hint - 5 : 0; 
unsigned last_nearby = _min (hint+5, L. count ()) ; 
for (int i = f irst_nearby; i < last_nearby; i++) { 
if (L[i] == S) return i; 
} 

// OK, give up on hint, try entire file. Must find it f 
for (int i = 0; i < L.countO; i++) 
if (L[i] =^ S) return i; 

printf ("INTERNAL ERROR! %s\n", functionO); 

} 

unsigned group_ix = 0; 

// For linked list of groups. 

struct Group { 

ExtendingLongVector start_position; 

Group ^next; 

Group 0 { next = 0; size = 0; } 

int size; 

}; 

Group *groups = 0, *last_group = 0; 

for S <- symbols_for_this_iteration. each__element ( ) do { 
if (occurs_uniquely (S) ) { 

// printf ("%d unique: %s\n", S, st . to___string (S) ) ; 

unsigned hint = syin_inf o [S] . line_index; 

if (absorbeds [input_count-l] .contains (hint) ) 

// This unique line has been absorbed in a previous group 
continue; 

// Find the position in each input where it occurs. 
//We have recorded its position in the last input, 
dbg && printf ( "--Focusing around unique line 
n", st .to_string(S) ) ; 

ExtendingLongVector start_position; 
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// Look at the last input and back up from there, because 
// the sym index was recorded for the last input stream. 
start_position [input_count-l] = hint; 

// We hope that it occurs here or nearby in the other files 

// If not search those files. 

for (int i = 0; i < input_count-l ; i++) { 

unsigned find_it = f ind_line_in (i, S, hint ) ; 

dbg print f ("unique line %s found at %d in stream %d\n" 
st . t o_str ing ( S ) , f ind_it , i) , 

start_position [i] = find_it; 

absorbeds[i] . add_element ( f ind_it ) ; 

} 

absorbeds [input_count-l] . add_element (hint ) ; 

Group *g = new Group () ; 

// Copy starting data. 

g. start_position = start_position; 

// Now go forward from S. Mark any lines found as not uniq 
// because we're absorbing them. 
Symbol next = Symbols : :NULL_SYMBOL; 
unsigned gsize = 0; 

void absorb_lines (ExtendingLongVector &start_position) { 
for (int i = 0; i < input_count; i++) { 

absorbeds [i] . add_element (start_position [i] ) ; 
// dbg &5c printf ( "absorb stream %d line 
%d\n" ^ i, start_position [i] ) ; 
} 

} 

while (1) { 

for (int i = 0; i < input_count; i++) { 

LongVector &L = *inf os [i] . symindex_vector; 
unsigned posn -i--i-start_position [i] ; 
if (posn > infos[i].hi) { 

// If we are at EOF, stop; no one can now match. 
#define STOP ( ) do {dbg && printf ("i %d reason 

%d\n",i, LINE ); goto FWD_STOP; } while (0) 

STOP ( ) ; 
} 

if (i == 0) { 
next = L [posn] ; 

dbg && printf ("next is %s\n"^ st . to_string (next )) ; 
} 

if (absorbeds [i] . contains (posn) ) { 
dbg && printf ("%d: line %d already 
absorbed. \n", i, posn) ; 

print_set (absorbeds [i] ) ; 

STOPO ; 

} 

else if (next L[posn]) ; 

else STOPO ; 

} 

// We made it through all the inputs. 
absorb_lines ( start_position) ; 
gsize++; 
} 

#undef STOP 
FWD STOP: ; 
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// Now back up and look at prior lines — i.e., expand in 
// the backwards direction. 
start_position = g. start__position; 
while (1) { 

for (int i ^ 0; i < input_count; i++) { 

LongVector &L = ^inf os [i] . symindex_veGtor ; 
int posn = — start__position [i] ; 
if (posn < infos[i].lo) { 

// If we went below, stop; no one can now match. 
#define STOP ( ) do {dbg printf("i %d reason 

%d\n",i, LINE ); goto BWD_STOP; } while (0) 

STOPO ; 
} 

if (i == 0) { 
next = L [posn] ; 

dbg && printf("next is %s\n" , st . to_string (next ) ) ; 
} 

if (absorbeds [i] , contains (posn) ) { 
dbg && printf ("%d: line %d already 
absorbed. \n", i,posn) ; 

STOP ( ) ; 
} 

else if (next L[posn]) ; 

else STOPO ; 

} 

// We made it through all the inputs. 
absorb_lines (start_position) ; 
gsi2e++; 

g . start_position = start_position; 
} 

#undef STOP 
BWD_STOP: ; 

gsi2e++; // Include the initial unique line. 

dbg &&printf ( "Group around this line is %d long , \n" , gsize) 

if (last_group == 0) groups = last_group = g; 

else last__group . next = g, last_group = g; 

last_group . size = gsize; 

if (dbg) { 

printf ( "Start positions for this group:\n"); 
for (int i = 0; i < input_count; i++) 

printf ("Id g. start_position [i] ) ; 
printf ("\n") ; 
} 

} 

} 

// Now ensure the groups are in the same sequence in all input 
streams . 
// 

for (Group *g = groups; g; ) { 
Group "^ng = g.next; 
if (ng == 0) break; 

// Check out the next group's starting position and compare to 
// this group* s. 

for (int i = 0; i < input_count; i++) { 

if (g . start_position [i] < ng. start_position [i] ) ; 
else { 
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dbg && printf ( "Sorry, out of sequence groups. Can't 
compare . \n" ) ; 

goto BAIL; 
} 

} 

g - ng; 

} 

void print_groups ( ) { 

// Print out the groups. 

for (Group *g = groups; g; g = g.next) { 

print f ( "Group length %d starts at ",g.size); 
for (int i = 0; i < input_count; i++) 

printf ( "%d " , g. start_position [i] ) ; 
printf ("\n") ; 
} 

} 

if (dbg) print_groups () ; 

void print_group_contents (Group *g) -> (OUT, OUT) { 
// Get the prefix for "all", 
unsigned first = 0, last = input_count-l ; 
void all_inputs() -> ( Input ^) { 

for (int j = first; j <= last; j++) yield (infos [j ], input ) 

} 

String prefix; 

inputs . identif ying_pref ix (all_inputs, prefix) ; 
for (int i = 0; i < g.size; i++) { 

// All the lines match among the various inputs, 
// so choose any input to print the contents, 
yield (prefix. to_string ( ) , 

st . to_string ( (*inf os [0] . symindex_vector ) 
[g . start_position [0] +i] ) ) ; 

} 

} 

if (groups) { 

// OK, we have groups* 

dbg && printf (" [%d] We have groups, so recursing. \n", depth) ; 
// First do the text up to the first group, if there is any. 
Input_infos R; bool found = FALSE; 
for (int i = 0; i < input_count; i++) { 

R[i] = infos [i]; // Copy other interesting data. 

R[i].lo - infos[i].lo; 

R[i] .hi = groups . start_position [i] -1 ; 

found = R[i].hi >= R[i].lo; 

} 

if (! found) { 

dbg && printf ( "Nothing leading up to the first group. \n") 
} 

else each_output (yield, R) ; 
for (Group *g = groups; g; ) { 
if (dbg) { 

printf ( " [%d] Group contents : \n", depth) ; 

for A,B <- print_group_contents (g) do printf ("\t%s 

%s\n",A,B) ; 

} 

print_group_contents (yield, g) ; 

// Do the text between this group and the next. 



-31- 



Group "^ng = g.next; 

bool this_is_last = ng == 0; 

Input_infos R; bool found = FALSE; 

for (int i = 0; i < input_count; i++) { 

R[i] = infos [i]; // Copy other stuff. 

R[i] .lo = g. start_position [i] +g, size; 

R[i] .hi = this_is_last ? infos [i], hi : ng . start_position [i] - 

1; 

// printf("i %d lo %d hi ld\n", i, R [i] . lo,R[i] .hi) ; 

if (R[i].hi >= R[i],lo) found - TRUE; 

} 

if (found) each_output (yield, R) ; 

else dbg printf ( "nothing after group of size %d\n'\ g. size) ; 

g = ng; 

} 

} 

else non_group_comparisons (yield, infos, maxlc) ; 
BAIL: ; 

// Clean up. 

for (int i - 0; i < input_count; i++) delete absorbeds [i] ; 

delete [] absorbeds; 

delete symbols_f or_this_iteration; 

void remove_groups (Group ^groups) { 

if (groups) remove_groups (groups .next) ; 

delete groups; 

} 

remove_groups (groups) ; 

depth — ; 

} 

void Real_aggregator : : non_group_comparisons ( 

Input_inf OS &infos, int maxlc) -> (OUT, OUT) { 
int input_count = inf os . count () ; 

// We get here if we have no groups. This reduces to a simple 
// line-by-line comparison. 

// Now compute the aggregate output from the individual inputs. 
// First compute the number of differences we'll have — i.e., lines 
// that are not identical across all inputs. Count not only 
differences, but 
// matches. 

dbg && printf ("max %d lines\n", maxlc) ; 
void combine_where_possible (Set *exclude=0) -> { 
Set ^contains, const char ^output) { 

// Here go line by line and combine corresponding matching lines 

// for the subset of streams for which they match. 

// Mark any stream that differs everywhere and print him 

// separately. As a degenerate case, we might print all 

// streams separately. 

Set Sprinted = Set : :make_Set ( ) , 
^contains = Set : :make_Set ( ) ; 

// For each line position, print it in a combined fashion. 

// Find each input having a line and print that line. 

/ / Combined output . 

dbg printf ( "maxlc %d\n" , maxlc) ; 

for (int line = 0; line < maxlc; line++) { 
printed.make_empty ( ) ; 
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// Now find out how many inputs have this particular line. 
// Some of them may be short. 
// Find a line and print it. 
int first ^ -1/ 
while (1) { 

Symbol S = Symbols :: NULL_SYMBOL; 

contains .make_empty ( ) ; 

for (int i = 0; i < input_count; i++) { 

if (exclude && exclude . contains (i ) ) continue; 
dbg ScSc printf("i %d\n",i); 

LongVector &vec = *inf os [i] . symindex_vector; 
int this_line = line + infos[i].lo; 
if (infos [i]. hi >= this_line) { 

Symbol candidate = vec [this_line] ; 
dbg && printf ("line[%d] = 
%s\n", i, st . to_string (candidate) ) ; 

if (S Symbols: :NULL_SYMBOL && 

! printed . contains ( candidate ) ) { 

S = candidate; 

} 

if (S == candidate) contains , add_element (i) ; 
} 

} 

if (S == Symbols: :NULL_SYMBOL) break; // nothing, 
else { 

yield (contains, st . to_string (S) ) ; 
// printf("add %d to printed\n" , S ) ; 
printed. add_element (S) ; 
} 

} 

} 

delete printed; 
delete contains; 
} 

// First determine who has matches and who doesn't. 
ExtendingLongVector match_count; // mat ch^count [stream] . 

for contains, output <- coxnbine_where_possible ( ) do { 
if (contains , count__elements ( ) > 1) 

for S <- contains . each__element ( ) do match_count [S] ++ ; 

} 

// Mark any stream with a low match count to be printed separately. 

Set ^separately = Set : :make_Set () ; 

for (int i = 0; i < input_count; i++) { 

// Your matches are computed based on your lines only, so if 
// another stream has substantially more lines than you, it 
doesn't change 

// your computation. So, the case A; A; B C will cause A and A 
// to be combined. 

int nlines = inf os [i] . hi-inf os [i] . lo+l ; 

unsigned diff_percent = nlines ? 100-100*match_count [i] /nlines : 

100; 

if (dif f__percent > tolerance) separately . add__element (i) ; 
} 

if (input_count - separately . count_elements ( ) > 0) { 

for contains, output <- combine_where_possible (separately) do { 
// Print the line for inputs first.. i. 
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string prefix; 

void the_inputs() -> (Input*) { 

for S <- contains . each_element ( ) do yield (infos [S] . input ) ; 

} 

inputs . identifying_pref ix (the_inputs^ prefix) ; 

yield (prefix . to_string ( ) , output ) ; 

} 



} 



for i <- separately . each_eleraent 0 do { 

// Distinct output; don't even try to combine. Too much mess, 
dbg && printf{ "distinct l\n") ; 
print_separately (yield, i, infos [i] ) ; 
} 

delete separately; 
} 

void Real_aggregator : :print_separately (int i^ Input_info &R) -> 
(OUT, OUT) { 

for (int line = R.lo; line <= R.hi; line++) { 

dbg && printf("rlo %d rhi %d tl %d\n" , R. lo, R. hi, line ) ; 
LongVector Stvec = *R. symindex_vector ; 
const char *text = st . to_string (vec [line] ) ; 
dbg printf ("line[%d] = %s\n" , i, text ) ; 
String prefix; 

void the_input ( ) -> (Input*) { yield (R. input ) ; } 
inputs . identif ying_pref ix (the_input, prefix) ; 
yield (prefix . to_string { ) , text ) ; 
} 

} 



#if TEST 

// Test example, 
tdefine CASE 2 

struct { char *A[3] ; } lines [] 
#if CASE — 1 



{"Z-1", 


"a=l", 


"2-1% }, 


{"A=l", 


"A=l", 


"A=l", }, 


{"R=2", 


"C=2", 


"C-2", }, 


{"B-3", 


"B=3", 


"B=3", }, 


{"C=4", 


"C=4", 


0, }, 


{"B-3", 


"B=3", 


"B=3% }, 


{"D=5", 


"D=5", 


"D=5% }, 


{"E=7", 


"E=7", 


"E=7", }, 


{"B=3", 


"B=3", 


"B=3% }, 


}; 






#elif CASE = 


= 2 




{"reg", 


"reg", 


"reg", 



// Gets absorbed with the D group. 

// Gets absorbed and marked non-unique. 



"disasm", "disasm", "disasm", 
"instrhist", 0, 0, 
"stack", "stack", "stack", 
"hwstack", 0, "hwstack", 
"globals", "globals", "globals", 
"funcs", "funcs", "funcs", 
"locals " , "locals " , "locals " , 
"source", "source", "source", 
"files", "files", "files". 
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{"brk", 
{"watch", 
{"mem", 
{ "examine", " 



"brk", 
"watch", 
"mem", 
examine' 



{"mod", 
}; 

#endif 



"mod", 



"brk", 
"watch", 
"mem", 
" examine 
"mod". 



struct Inp: Aggregator :: Input { 

Inp(int which) { this. which = which; } 
unsigned which; 

static const int max = sizeof ( lines ) /sizeof ( *lines ) ; 
virtual void each_line() -> (const char^line, char = TRQE) { 
for (int i = 0; i < max; i++) { 

char *p = lines [i] .A[which-1] ; 

if (p) yield (p); 

} 

}; 

}; 

struct Inps : Aggregator ;: Inputs { 
InpsO : A(1),B(2),C(3) { 

const int max = sizeof (lines) /sizeof (*lines) ; 

int lent [3] = {0,0,0}; 

for (int i = 0; i < max; i++) { 

for (int which = 1; which <= 3; which++) { 
char *p = lines [i] .A[which-1] ; 
if (p) printf(" %d : %s", lent [which-1] ++, p) ; 
else printf(" -none-"); 
printf ("\t") ; 
} 

printf ("\n") ; 
} 

} 

Inp A, B, C; 

virtual void each___input ( ) -> (Aggregator :: Input ) { 
yield(&A); yield(&B); yield(&C); 
} 

// Compute the line prefix given the set of inputs, 
virtual void identif ying_pref ix ( 

void each_input 0 -> (Aggregator :: Input*) ! , String &s) { 
int cnt = 0; 

for I <- each_input() do cnt++; 

if (cnt ==0) s - ""; // Shouldn't happen. 

else if (cnt 3) s = "[all]"; 

else { 

s - ' [ ' ; 

for I <- each_input() do s.aprintf(" %d", ( (Inp* ) I ) . which) 

s += n ^ 

} 

} 



main () { 
Inps I; 

Aggregator *A = Aggregator : :make_Aggregator () ; 
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A. set_tolerance (50) ; 
for P,X <- A.each_output (&I) do 
printf("%s: %s\n",P,X); 

} 

5 #endif 



Eg., 

10 globals 

eval X (some array, say) 
For CL display, we can run the output for each, aggregate it, 
do a differential comparator, and then produce the result. 
E.g. for process 1: 
15 [1] A = 3 

[1] B = 2 
[1] C = 1 
Process 2: 

[2] A = 3 
20 [2] B = 5 

[2] C - 1 
We should print 
[1.2] A = 3 
[1] B = 2 
25 [2] B = 5 

[1,2] C - 1 

Or, 

[*] A = 3 
[1] B - 2 
30 [2] B = 5 

[^] C = 1 



Recursive algorithm: 

35 

void diffs(void each_stream) { 

enter each line of the stream in a string table and assign 

each unique line a unique string index; 
Identify unique lines that are present in each file; 
40 Go forward and backwards from each unique line to broaden 

regions of identical lines; 
Check that these regions are in the same order for all streams. 
If they are not, use the line-by-line algorithm and return. 
For R <- each region 
45 yield each region as matched by all streams 

recursively process the text between this region and the next 

} 

50 V 
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